#
# Copyright 2008-2010 Zuza Software Foundation
#
# This file is part of translate.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, see <http://www.gnu.org/licenses/>.

"""
A translation memory server using tmdb for storage, communicates with
clients using JSON over HTTP.
"""

import json
import logging
from argparse import ArgumentParser
from io import BytesIO
from urllib import parse

from translate.misc import selector, wsgi
from translate.storage import base, tmdb

logger = logging.getLogger(__name__)


class TMServer:
    """A RESTful JSON TM server."""

    def __init__(
        self,
        tmdbfile,
        tmfiles,
        max_candidates=3,
        min_similarity=75,
        max_length=1000,
        prefix="",
        source_lang=None,
        target_lang=None,
    ):
        if not isinstance(tmdbfile, str):
            import sys

            tmdbfile = tmdbfile.decode(sys.getfilesystemencoding())

        self.tmdb = tmdb.TMDB(tmdbfile, max_candidates, min_similarity, max_length)

        if tmfiles:
            self._load_files(tmfiles, source_lang, target_lang)

        # initialize url dispatcher
        self.rest = selector.Selector(prefix=prefix)
        self.rest.add(
            "/{slang}/{tlang}/unit/{uid:any}",
            GET=self.translate_unit,
            POST=self.update_unit,
            PUT=self.add_unit,
            DELETE=self.forget_unit,
        )

        self.rest.add(
            "/{slang}/{tlang}/store/{sid:any}",
            GET=self.get_store_stats,
            PUT=self.upload_store,
            POST=self.add_store,
            DELETE=self.forget_store,
        )

    def _load_files(self, tmfiles, source_lang, target_lang):
        from translate.storage import factory

        if isinstance(tmfiles, list):
            for tmfile in tmfiles:
                self.tmdb.add_store(factory.getobject(tmfile), source_lang, target_lang)
        elif tmfiles:
            self.tmdb.add_store(factory.getobject(tmfiles), source_lang, target_lang)

    @selector.opliant
    def translate_unit(self, environ, start_response, uid, slang, tlang):
        start_response("200 OK", [("Content-type", "text/plain")])
        candidates = self.tmdb.translate_unit(uid, slang, tlang)
        logger.debug("candidates: %s", str(candidates))
        response = json.dumps(candidates, indent=4).encode("utf-8")
        params = parse.parse_qs(environ.get("QUERY_STRING", ""))
        try:
            callback = params.get("callback", [])[0]
            response = f"{callback}({response})"
        except IndexError:
            pass
        return [response]

    @selector.opliant
    def add_unit(self, environ, start_response, uid, slang, tlang):
        start_response("200 OK", [("Content-type", "text/plain")])
        # uid = unicode(urllib.unquote_plus(uid), "utf-8")
        data = json.loads(environ["wsgi.input"].read(int(environ["CONTENT_LENGTH"])))
        unit = base.TranslationUnit(data["source"])
        unit.target = data["target"]
        self.tmdb.add_unit(unit, slang, tlang)
        return [""]

    @selector.opliant
    def update_unit(self, environ, start_response, uid, slang, tlang):
        start_response("200 OK", [("Content-type", "text/plain")])
        # uid = unicode(urllib.unquote_plus(uid), "utf-8")
        data = json.loads(environ["wsgi.input"].read(int(environ["CONTENT_LENGTH"])))
        unit = base.TranslationUnit(data["source"])
        unit.target = data["target"]
        self.tmdb.add_unit(unit, slang, tlang)
        return [""]

    @selector.opliant
    def forget_unit(self, environ, start_response, uid):
        # FIXME: implement me
        start_response("200 OK", [("Content-type", "text/plain")])
        # uid = unicode(urllib.unquote_plus(uid), "utf-8")

        return ["FIXME"]

    @selector.opliant
    def get_store_stats(self, environ, start_response, sid):
        # FIXME: implement me
        start_response("200 OK", [("Content-type", "text/plain")])
        # sid = unicode(urllib.unquote_plus(sid), "utf-8")

        return ["FIXME"]

    @selector.opliant
    def upload_store(self, environ, start_response, sid, slang, tlang):
        """Add units from uploaded file to tmdb."""
        from translate.storage import factory

        start_response("200 OK", [("Content-type", "text/plain")])
        data = BytesIO(environ["wsgi.input"].read(int(environ["CONTENT_LENGTH"])))
        data.name = sid
        store = factory.getobject(data)
        count = self.tmdb.add_store(store, slang, tlang)
        response = "added %d units from %s" % (count, sid)
        return [response]

    @selector.opliant
    def add_store(self, environ, start_response, sid, slang, tlang):
        """Add unit from POST data to tmdb."""
        start_response("200 OK", [("Content-type", "text/plain")])
        units = json.loads(environ["wsgi.input"].read(int(environ["CONTENT_LENGTH"])))
        count = self.tmdb.add_list(units, slang, tlang)
        response = "added %d units from %s" % (count, sid)
        return [response]

    @selector.opliant
    def forget_store(self, environ, start_response, sid):
        # FIXME: implement me
        start_response("200 OK", [("Content-type", "text/plain")])
        # sid = unicode(urllib.unquote_plus(sid), "utf-8")

        return ["FIXME"]


def main():
    parser = ArgumentParser()
    parser.add_argument(
        "-d",
        "--tmdb",
        dest="tmdbfile",
        default=":memory:",
        help="translation memory database file",
    )
    parser.add_argument(
        "-f",
        "--import-translation-file",
        dest="tmfiles",
        action="append",
        help="translation file to import into the database",
    )
    parser.add_argument(
        "-t",
        "--import-target-lang",
        dest="target_lang",
        help="target language of translation files",
    )
    parser.add_argument(
        "-s",
        "--import-source-lang",
        dest="source_lang",
        help="source language of translation files",
    )
    parser.add_argument(
        "-b",
        "--bind",
        dest="bind",
        default="localhost",
        help="address to bind server to (default: %(default)s)",
    )
    parser.add_argument(
        "-p",
        "--port",
        dest="port",
        type=int,
        default=8888,
        help="port to listen on (default: %(default)s)",
    )
    parser.add_argument(
        "--max-candidates",
        dest="max_candidates",
        type=int,
        default=3,
        help="Maximum number of candidates",
    )
    parser.add_argument(
        "--min-similarity",
        dest="min_similarity",
        type=int,
        default=75,
        help="minimum similarity",
    )
    parser.add_argument(
        "--max-length",
        dest="max_length",
        type=int,
        default=1000,
        help="Maxmimum string length",
    )
    parser.add_argument(
        "--debug",
        action="store_true",
        dest="debug",
        default=False,
        help="enable debugging features",
    )

    args = parser.parse_args()

    # setup debugging
    format = "%(asctime)s %(levelname)s %(message)s"
    level = logging.DEBUG if args.debug else logging.WARNING
    if args.debug:
        format = "%(levelname)7s %(module)s.%(funcName)s:%(lineno)d: %(message)s"

    logging.basicConfig(level=level, format=format)

    application = TMServer(
        args.tmdbfile,
        args.tmfiles,
        max_candidates=args.max_candidates,
        min_similarity=args.min_similarity,
        max_length=args.max_length,
        prefix="/tmserver",
        source_lang=args.source_lang,
        target_lang=args.target_lang,
    )
    wsgi.launch_server(args.bind, args.port, application.rest)


if __name__ == "__main__":
    main()
